"""Test config flow."""

from ipaddress import ip_address
import json
from typing import Any
from unittest.mock import AsyncMock, MagicMock, patch

from aioesphomeapi import (
    APIClient,
    APIConnectionError,
    DeviceInfo,
    InvalidAuthAPIError,
    InvalidEncryptionKeyAPIError,
    RequiresEncryptionAPIError,
    ResolveAPIError,
    wifi_mac_to_bluetooth_mac,
)
import aiohttp
import pytest

from homeassistant import config_entries
from homeassistant.components.esphome import dashboard
from homeassistant.components.esphome.const import (
    CONF_ALLOW_SERVICE_CALLS,
    CONF_DEVICE_NAME,
    CONF_NOISE_PSK,
    CONF_SUBSCRIBE_LOGS,
    DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
    DOMAIN,
)
from homeassistant.components.esphome.encryption_key_storage import (
    ENCRYPTION_KEY_STORAGE_KEY,
)
from homeassistant.config_entries import SOURCE_IGNORE, ConfigFlowResult
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_PORT
from homeassistant.core import HomeAssistant
from homeassistant.data_entry_flow import FlowResultType
from homeassistant.helpers import discovery_flow
from homeassistant.helpers.service_info.dhcp import DhcpServiceInfo
from homeassistant.helpers.service_info.esphome import ESPHomeServiceInfo
from homeassistant.helpers.service_info.hassio import HassioServiceInfo
from homeassistant.helpers.service_info.mqtt import MqttServiceInfo
from homeassistant.helpers.service_info.zeroconf import ZeroconfServiceInfo

from . import VALID_NOISE_PSK
from .conftest import MockGenericDeviceEntryType

from tests.common import MockConfigEntry


async def test_retrieve_encryption_key_from_storage_with_device_mac(
    hass: HomeAssistant,
    mock_client: APIClient,
    hass_storage: dict[str, Any],
) -> None:
    """Test key successfully retrieved from storage."""

    # Mock the encryption key storage
    hass_storage[ENCRYPTION_KEY_STORAGE_KEY] = {
        "version": 1,
        "minor_version": 1,
        "key": ENCRYPTION_KEY_STORAGE_KEY,
        "data": {"keys": {"11:22:33:44:55:aa": VALID_NOISE_PSK}},
    }

    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test", "11:22:33:44:55:AA"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )
    await hass.async_block_till_done()

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }

    assert mock_client.noise_psk == VALID_NOISE_PSK


async def test_reauth_fixed_from_from_storage(
    hass: HomeAssistant,
    mock_client: APIClient,
    hass_storage: dict[str, Any],
) -> None:
    """Test reauth fixed automatically via storage."""

    # Mock the encryption key storage
    hass_storage[ENCRYPTION_KEY_STORAGE_KEY] = {
        "version": 1,
        "minor_version": 1,
        "key": ENCRYPTION_KEY_STORAGE_KEY,
        "data": {"keys": {"11:22:33:44:55:aa": VALID_NOISE_PSK}},
    }

    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )

    result = await entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.ABORT, result
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


async def test_retrieve_encryption_key_from_storage_no_key_found(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test _retrieve_encryption_key_from_storage when no key is found."""

    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )

    result = await entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.FORM, result
    assert result["step_id"] == "reauth_confirm"
    assert CONF_NOISE_PSK not in entry.data


INVALID_NOISE_PSK = "lSYBYEjQI1bVL8s2Vask4YytGMj1f1epNtmoim2yuTM="
WRONG_NOISE_PSK = "GP+ciK+nVfTQ/gcz6uOdS+oKEdJgesU+jeu8Ssj2how="


@pytest.fixture(autouse=False)
def mock_setup_entry():
    """Mock setting up a config entry."""
    with patch("homeassistant.components.esphome.async_setup_entry", return_value=True):
        yield


def get_flow_context(hass: HomeAssistant, result: ConfigFlowResult) -> dict[str, Any]:
    """Get the flow context from the result of async_init or async_configure."""
    flow = next(
        flow
        for flow in hass.config_entries.flow.async_progress()
        if flow["flow_id"] == result["flow_id"]
    )

    return flow["context"]


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_connection_works(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test we can finish a config flow."""
    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 80},
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 80,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }
    assert result["options"] == {
        CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS
    }
    assert result["title"] == "test"
    assert result["result"].unique_id == "11:22:33:44:55:aa"

    assert len(mock_client.connect.mock_calls) == 1
    assert len(mock_client.device_info.mock_calls) == 1
    assert len(mock_client.disconnect.mock_calls) == 1
    assert mock_client.host == "127.0.0.1"
    assert mock_client.port == 80
    assert mock_client.password == ""
    assert mock_client.noise_psk is None


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_user_connection_updates_host(hass: HomeAssistant) -> None:
    """Test setup up the same name updates the host."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "test.local", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data=None,
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 80},
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_updates"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }
    assert entry.data[CONF_HOST] == "127.0.0.1"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_user_sets_unique_id(hass: HomeAssistant) -> None:
    """Test that the user flow sets the unique id."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
        },
        type="mock_type",
    )
    discovery_result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert discovery_result["type"] is FlowResultType.FORM
    assert discovery_result["step_id"] == "discovery_confirm"
    assert discovery_result["description_placeholders"] == {
        "name": "test8266",
    }

    discovery_result = await hass.config_entries.flow.async_configure(
        discovery_result["flow_id"],
        {},
    )
    assert discovery_result["type"] is FlowResultType.CREATE_ENTRY
    assert discovery_result["data"] == {
        CONF_HOST: "192.168.43.183",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data=None,
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        {CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_updates"
    assert result["description_placeholders"] == {
        "title": "test",
        "name": "test",
        "mac": "11:22:33:44:55:aa",
    }


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_resolve_error(hass: HomeAssistant, mock_client: APIClient) -> None:
    """Test user step with IP resolve error."""

    with patch(
        "homeassistant.components.esphome.config_flow.APIConnectionError",
        new_callable=lambda: ResolveAPIError,
    ) as exc:
        mock_client.device_info.side_effect = exc
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
        )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"
    assert result["errors"] == {"base": "resolve_error"}

    assert len(mock_client.connect.mock_calls) == 1
    assert len(mock_client.device_info.mock_calls) == 1
    assert len(mock_client.disconnect.mock_calls) == 1

    # Now simulate the user retrying with the same host and a successful connection
    mock_client.device_info.side_effect = None

    result2 = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result2["type"] == FlowResultType.CREATE_ENTRY
    assert result2["title"] == "test"
    assert result2["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_DEVICE_NAME: "test",
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
    }


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_user_causes_zeroconf_to_abort(hass: HomeAssistant) -> None:
    """Test that the user flow sets the unique id and aborts the zeroconf flow."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
        },
        type="mock_type",
    )
    discovery_result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert discovery_result["type"] is FlowResultType.FORM
    assert discovery_result["step_id"] == "discovery_confirm"
    assert discovery_result["description_placeholders"] == {
        "name": "test8266",
    }

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data=None,
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        {CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }

    assert not hass.config_entries.flow.async_progress_by_handler(DOMAIN)


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_connection_error(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test user step with connection error."""
    mock_client.device_info.side_effect = APIConnectionError

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "user"
    assert result["errors"] == {"base": "connection_error"}

    assert len(mock_client.connect.mock_calls) == 1
    assert len(mock_client.device_info.mock_calls) == 1
    assert len(mock_client.disconnect.mock_calls) == 1

    # Now simulate the user retrying with the same host and a successful connection
    mock_client.device_info.side_effect = None

    result2 = await hass.config_entries.flow.async_configure(
        result["flow_id"],
        user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result2["type"] == FlowResultType.CREATE_ENTRY
    assert result2["title"] == "test"
    assert result2["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_DEVICE_NAME: "test",
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
    }


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_with_password(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test user step with password."""
    mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "password1"}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "password1",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.password == "password1"


@pytest.mark.usefixtures("mock_zeroconf")
async def test_user_invalid_password(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test user step with invalid password."""
    mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {"name": "test"}

    mock_client.connect.side_effect = InvalidAuthAPIError

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "invalid"}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {"name": "test"}
    assert result["errors"] == {"base": "invalid_auth"}

    mock_client.connect.side_effect = None

    result2 = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "good"}
    )

    assert result2["type"] == FlowResultType.CREATE_ENTRY
    assert result2["title"] == "test"
    assert result2["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_DEVICE_NAME: "test",
        CONF_PASSWORD: "good",
        CONF_NOISE_PSK: "",
    }


@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
async def test_user_dashboard_has_wrong_key(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test user step with key from dashboard that is incorrect."""
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=WRONG_NOISE_PSK,
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_discovers_name_and_gets_key_from_dashboard(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test user step can discover the name and get the key from the dashboard."""
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )
    await dashboard.async_get_dashboard(hass).async_refresh()

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.parametrize(
    "dashboard_exception",
    [aiohttp.ClientError(), json.JSONDecodeError("test", "test", 0)],
)
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_discovers_name_and_gets_key_from_dashboard_fails(
    hass: HomeAssistant,
    dashboard_exception: Exception,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test user step can discover the name and get the key from the dashboard."""
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )
    await dashboard.async_get_dashboard(hass).async_refresh()

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        side_effect=dashboard_exception,
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_discovers_name_and_dashboard_is_unavailable(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test user step can discover the name but the dashboard is unavailable."""
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_devices",
        side_effect=TimeoutError,
    ):
        await dashboard.async_get_dashboard(hass).async_refresh()
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_login_connection_error(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test user step with connection error on login attempt."""
    mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {"name": "test"}

    mock_client.connect.side_effect = APIConnectionError

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "valid"}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {"name": "test"}
    assert result["errors"] == {"base": "connection_error"}

    mock_client.connect.side_effect = None

    result2 = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "good"}
    )

    assert result2["type"] == FlowResultType.CREATE_ENTRY
    assert result2["title"] == "test"
    assert result2["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_DEVICE_NAME: "test",
        CONF_PASSWORD: "good",
        CONF_NOISE_PSK: "",
    }


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_initiation(hass: HomeAssistant) -> None:
    """Test discovery importing works."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
            "friendly_name": "The Test",
        },
        type="mock_type",
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )
    assert get_flow_context(hass, flow) == {
        "source": config_entries.SOURCE_ZEROCONF,
        "title_placeholders": {"name": "The Test (test)"},
        "unique_id": "11:22:33:44:55:aa",
    }

    result = await hass.config_entries.flow.async_configure(
        flow["flow_id"], user_input={}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test"
    assert result["data"][CONF_HOST] == "192.168.43.183"
    assert result["data"][CONF_PORT] == 6053

    assert result["result"]
    assert result["result"].unique_id == "11:22:33:44:55:aa"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_no_mac(hass: HomeAssistant) -> None:
    """Test discovery aborted if old ESPHome without mac in zeroconf."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={},
        type="mock_type",
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )
    assert flow["type"] is FlowResultType.ABORT
    assert flow["reason"] == "mdns_missing_mac"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_already_configured(hass: HomeAssistant) -> None:
    """Test discovery aborts if already configured via hostname."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "test8266.local", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )

    entry.add_to_hass(hass)

    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={"mac": "1122334455aa"},
        type="mock_type",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_updates"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_ignored(hass: HomeAssistant) -> None:
    """Test discovery does not probe and ignored entry."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "test8266.local", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
        source=SOURCE_IGNORE,
    )

    entry.add_to_hass(hass)

    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={"mac": "1122334455aa"},
        type="mock_type",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_duplicate_data(hass: HomeAssistant) -> None:
    """Test discovery aborts if same mDNS packet arrives."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test.local.",
        name="mock_name",
        port=6053,
        properties={"address": "test.local", "mac": "1122334455aa"},
        type="mock_type",
    )

    result = await hass.config_entries.flow.async_init(
        DOMAIN, data=service_info, context={"source": config_entries.SOURCE_ZEROCONF}
    )
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "discovery_confirm"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_init(
        DOMAIN, data=service_info, context={"source": config_entries.SOURCE_ZEROCONF}
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_in_progress"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_updates_unique_id(hass: HomeAssistant) -> None:
    """Test a duplicate discovery host aborts and updates existing entry."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )

    entry.add_to_hass(hass)

    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.184"),
        ip_addresses=[ip_address("192.168.43.184")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={"address": "test8266.local", "mac": "1122334455aa"},
        type="mock_type",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_updates"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }

    assert entry.data[CONF_HOST] == "192.168.43.184"
    assert entry.unique_id == "11:22:33:44:55:aa"


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_abort_without_update_same_host_port(
    hass: HomeAssistant,
) -> None:
    """Test discovery aborts without update when hsot and port are the same."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )

    entry.add_to_hass(hass)

    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={"address": "test8266.local", "mac": "1122334455aa"},
        type="mock_type",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_requires_psk(hass: HomeAssistant, mock_client: APIClient) -> None:
    """Test user step with requiring encryption key."""
    mock_client.device_info.side_effect = RequiresEncryptionAPIError

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["errors"] == {}
    assert result["description_placeholders"] == {"name": "ESPHome"}

    assert len(mock_client.connect.mock_calls) == 2
    assert len(mock_client.device_info.mock_calls) == 2
    assert len(mock_client.disconnect.mock_calls) == 2

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
    )
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["errors"] == {"base": "requires_encryption_key"}
    assert result["description_placeholders"] == {"name": "ESPHome"}

    mock_client.device_info.side_effect = None
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_encryption_key_valid_psk(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test encryption key step with valid key."""

    mock_client.device_info.side_effect = RequiresEncryptionAPIError

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "ESPHome"}

    device_info = DeviceInfo(uses_password=False, name="test")
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_encryption_key_invalid_psk(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test encryption key step with invalid key."""

    mock_client.device_info.side_effect = RequiresEncryptionAPIError

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "ESPHome"}

    mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["errors"] == {"base": "invalid_psk"}
    assert result["description_placeholders"] == {"name": "ESPHome"}
    assert mock_client.noise_psk == INVALID_NOISE_PSK

    mock_client.device_info.side_effect = None
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_confirm_valid(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth initiation with valid PSK."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reauth_flow(hass)
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "reauth_confirm"
    assert result["description_placeholders"] == {
        "name": "Mock Title (test)",
    }

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reauth_attempt_to_change_mac_aborts(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth initiation with valid PSK attempting to change mac.

    This can happen if reauth starts, but they don't finish it before
    a new device takes the place of the old one at the same IP.
    """
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reauth_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_unique_id_changed"
    assert CONF_NOISE_PSK not in entry.data
    assert result["description_placeholders"] == {
        "expected_mac": "11:22:33:44:55:aa",
        "host": "127.0.0.1",
        "name": "test",
        "unexpected_device_name": "test",
        "unexpected_mac": "11:22:33:44:55:bb",
    }


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reauth_password_changed(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth when password has changed."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: "old_password"},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    mock_client.connect.side_effect = InvalidAuthAPIError("Invalid password")

    result = await entry.start_reauth_flow(hass)
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "authenticate"
    assert result["description_placeholders"] == {
        "name": "Mock Title",
    }

    mock_client.connect.side_effect = None
    mock_client.connect.return_value = None
    mock_client.device_info.return_value = DeviceInfo(
        uses_password=True, name="test", mac_address="11:22:33:44:55:aa"
    )

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_PASSWORD: "new_password"}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_PASSWORD] == "new_password"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_fixed_via_dashboard(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test reauth fixed automatically via dashboard."""

    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )

    await dashboard.async_get_dashboard(hass).async_refresh()

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ) as mock_get_encryption_key:
        result = await entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.ABORT, result
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK

    assert len(mock_get_encryption_key.mock_calls) == 1


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_fixed_via_dashboard_add_encryption_remove_password(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
    mock_config_entry: MockConfigEntry,
) -> None:
    """Test reauth fixed automatically via dashboard with password removed."""
    mock_client.device_info.side_effect = (
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(uses_password=False, name="test", mac_address="11:22:33:44:55:aa"),
    )

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )

    await dashboard.async_get_dashboard(hass).async_refresh()

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ) as mock_get_encryption_key:
        result = await mock_config_entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.ABORT, result
    assert result["reason"] == "reauth_successful"
    assert mock_config_entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
    assert mock_config_entry.data[CONF_PASSWORD] == ""

    assert len(mock_get_encryption_key.mock_calls) == 1


@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
async def test_reauth_fixed_via_remove_password(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_config_entry: MockConfigEntry,
) -> None:
    """Test reauth fixed automatically by seeing password removed."""
    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )

    result = await mock_config_entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.ABORT, result
    assert result["reason"] == "reauth_successful"
    assert mock_config_entry.data[CONF_PASSWORD] == ""


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_fixed_via_dashboard_at_confirm(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test reauth fixed automatically via dashboard at confirm step."""

    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )

    result = await entry.start_reauth_flow(hass)

    assert result["type"] is FlowResultType.FORM, result
    assert result["step_id"] == "reauth_confirm"
    assert result["description_placeholders"] == {
        "name": "Mock Title (test)",
    }

    mock_dashboard["configured"].append(
        {
            "name": "test",
            "configuration": "test.yaml",
        }
    )

    await dashboard.async_get_dashboard(hass).async_refresh()

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ) as mock_get_encryption_key:
        # We just fetch the form
        result = await hass.config_entries.flow.async_configure(result["flow_id"])

    assert result["type"] is FlowResultType.ABORT, result
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK

    assert len(mock_get_encryption_key.mock_calls) == 1


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_confirm_invalid(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth initiation with invalid PSK."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reauth_flow(hass)

    mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "reauth_confirm"
    assert result["description_placeholders"] == {
        "name": "Mock Title (test)",
    }
    assert result["errors"]
    assert result["errors"]["base"] == "invalid_psk"

    device_info = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_reauth_confirm_invalid_with_unique_id(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth initiation with invalid PSK."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reauth_flow(hass)

    mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "reauth_confirm"
    assert result["description_placeholders"] == {
        "name": "Mock Title (test)",
    }
    assert result["errors"]
    assert result["errors"]["base"] == "invalid_psk"

    device_info = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_reauth_encryption_key_removed(hass: HomeAssistant) -> None:
    """Test reauth when the encryption key was removed."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_NOISE_PSK: VALID_NOISE_PSK,
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reauth_flow(hass)
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "reauth_encryption_removed_confirm"
    assert result["description_placeholders"] == {
        "name": "Mock Title (test)",
    }

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_successful"
    assert entry.data[CONF_NOISE_PSK] == ""


async def test_reauth_different_device_at_same_address(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reauth aborts when a different device is found at the same IP address."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_NOISE_PSK: VALID_NOISE_PSK,
            CONF_DEVICE_NAME: "old_device",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    # Mock a different device at the same IP (different MAC address)
    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False,
        name="new_device",
        legacy_bluetooth_proxy_version=0,
        # Different MAC address than the entry
        mac_address="AA:BB:CC:DD:EE:FF",
        esphome_version="1.0.0",
    )

    result = await entry.start_reauth_flow(hass)
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reauth_unique_id_changed"
    assert result["description_placeholders"] == {
        "name": "old_device",
        "host": "127.0.0.1",
        "expected_mac": "11:22:33:44:55:aa",
        "unexpected_mac": "aa:bb:cc:dd:ee:ff",
        "unexpected_device_name": "new_device",
    }


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_discovery_dhcp_updates_host(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test dhcp discovery updates host and aborts."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    device_info = DeviceInfo(name="test8266", mac_address="1122334455aa")
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )

    service_info = DhcpServiceInfo(
        ip="192.168.43.184",
        hostname="test8266",
        macaddress="1122334455aa",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_updates"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }

    assert entry.data[CONF_HOST] == "192.168.43.184"


@pytest.mark.usefixtures("mock_setup_entry")
async def test_discovery_dhcp_does_not_update_host_wrong_mac(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test dhcp discovery does not update the host if the mac is wrong."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    device_info = DeviceInfo(name="test8266", mac_address="1122334455ff")
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )

    service_info = DhcpServiceInfo(
        ip="192.168.43.184",
        hostname="test8266",
        macaddress="1122334455aa",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_detailed"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }

    # Mac was wrong, should not update
    assert entry.data[CONF_HOST] == "192.168.43.183"


@pytest.mark.usefixtures("mock_setup_entry")
async def test_discovery_dhcp_does_not_update_host_wrong_mac_bad_key(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test dhcp discovery does not update the host if the mac is wrong."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError(
        "Wrong key", "test8266", "1122334455cc"
    )
    service_info = DhcpServiceInfo(
        ip="192.168.43.184",
        hostname="test8266",
        macaddress="1122334455aa",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_detailed"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }

    # Mac was wrong, should not update
    assert entry.data[CONF_HOST] == "192.168.43.183"


@pytest.mark.usefixtures("mock_setup_entry")
async def test_discovery_dhcp_does_not_update_host_missing_mac_bad_key(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test dhcp discovery does not update the host if the mac is missing."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError(
        "Wrong key", "test8266", None
    )
    service_info = DhcpServiceInfo(
        ip="192.168.43.184",
        hostname="test8266",
        macaddress="1122334455aa",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured_detailed"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "unknown",
        "mac": "11:22:33:44:55:aa",
    }

    # Mac was missing, should not update
    assert entry.data[CONF_HOST] == "192.168.43.183"


@pytest.mark.usefixtures("mock_setup_entry")
async def test_discovery_dhcp_no_changes(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test dhcp discovery updates host and aborts."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
    )
    entry.add_to_hass(hass)

    device_info = DeviceInfo(name="test8266")
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )

    service_info = DhcpServiceInfo(
        ip="192.168.43.183",
        hostname="test8266",
        macaddress="000000000000",
    )
    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured"

    assert entry.data[CONF_HOST] == "192.168.43.183"


@pytest.mark.usefixtures("mock_dashboard")
async def test_discovery_hassio(hass: HomeAssistant) -> None:
    """Test dashboard discovery."""
    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        data=HassioServiceInfo(
            config={
                "host": "mock-esphome",
                "port": 6052,
            },
            name="ESPHome",
            slug="mock-slug",
            uuid="1234",
        ),
        context={"source": config_entries.SOURCE_HASSIO},
    )
    assert result
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "service_received"

    dash = dashboard.async_get_dashboard(hass)
    assert dash is not None
    assert dash.addon_slug == "mock-slug"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_zeroconf_encryption_key_via_dashboard(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test encryption key retrieved from dashboard."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
        },
        type="mock_type",
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert flow["type"] is FlowResultType.FORM
    assert flow["step_id"] == "discovery_confirm"
    assert flow["description_placeholders"] == {"name": "test8266"}

    mock_dashboard["configured"].append(
        {
            "name": "test8266",
            "configuration": "test8266.yaml",
        }
    )

    await dashboard.async_get_dashboard(hass).async_refresh()

    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        DeviceInfo(
            uses_password=False,
            name="test8266",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ) as mock_get_encryption_key:
        result = await hass.config_entries.flow.async_configure(
            flow["flow_id"], user_input={}
        )

    assert len(mock_get_encryption_key.mock_calls) == 1

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test8266"
    assert result["data"][CONF_HOST] == "192.168.43.183"
    assert result["data"][CONF_PORT] == 6053
    assert result["data"][CONF_NOISE_PSK] == VALID_NOISE_PSK

    assert result["result"]
    assert result["result"].unique_id == "11:22:33:44:55:aa"

    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_zeroconf_encryption_key_via_dashboard_with_api_encryption_prop(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_dashboard: dict[str, Any],
) -> None:
    """Test encryption key retrieved from dashboard with api_encryption property set."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
            "api_encryption": "any",
        },
        type="mock_type",
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert flow["type"] is FlowResultType.FORM
    assert flow["step_id"] == "discovery_confirm"
    assert flow["description_placeholders"] == {"name": "test8266"}

    mock_dashboard["configured"].append(
        {
            "name": "test8266",
            "configuration": "test8266.yaml",
        }
    )

    await dashboard.async_get_dashboard(hass).async_refresh()

    mock_client.device_info.side_effect = [
        DeviceInfo(
            uses_password=False,
            name="test8266",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    with patch(
        "homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
        return_value=VALID_NOISE_PSK,
    ) as mock_get_encryption_key:
        result = await hass.config_entries.flow.async_configure(
            flow["flow_id"], user_input={}
        )

    assert len(mock_get_encryption_key.mock_calls) == 1

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test8266"
    assert result["data"][CONF_HOST] == "192.168.43.183"
    assert result["data"][CONF_PORT] == 6053
    assert result["data"][CONF_NOISE_PSK] == VALID_NOISE_PSK

    assert result["result"]
    assert result["result"].unique_id == "11:22:33:44:55:aa"

    assert mock_client.noise_psk == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
async def test_zeroconf_no_encryption_key_via_dashboard(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test encryption key not retrieved from dashboard."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "1122334455aa",
        },
        type="mock_type",
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
    )

    assert flow["type"] is FlowResultType.FORM
    assert flow["step_id"] == "discovery_confirm"
    assert flow["description_placeholders"] == {"name": "test8266"}

    await dashboard.async_get_dashboard(hass).async_refresh()

    mock_client.device_info.side_effect = RequiresEncryptionAPIError

    result = await hass.config_entries.flow.async_configure(
        flow["flow_id"], user_input={}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "test8266"}

    mock_client.device_info.side_effect = None
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "192.168.43.183",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


async def test_option_flow_allow_service_calls(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_generic_device_entry: MockGenericDeviceEntryType,
) -> None:
    """Test config flow options for allow service calls."""
    entry = await mock_generic_device_entry(
        mock_client=mock_client,
    )

    result = await hass.config_entries.options.async_init(entry.entry_id)
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "init"
    assert result["data_schema"]({}) == {
        CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
        CONF_SUBSCRIBE_LOGS: False,
    }

    result = await hass.config_entries.options.async_init(entry.entry_id)
    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "init"
    assert result["data_schema"]({}) == {
        CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
        CONF_SUBSCRIBE_LOGS: False,
    }
    with patch(
        "homeassistant.components.esphome.async_setup_entry", return_value=True
    ) as mock_reload:
        result = await hass.config_entries.options.async_configure(
            result["flow_id"],
            user_input={CONF_ALLOW_SERVICE_CALLS: True, CONF_SUBSCRIBE_LOGS: False},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_ALLOW_SERVICE_CALLS: True,
        CONF_SUBSCRIBE_LOGS: False,
    }
    assert len(mock_reload.mock_calls) == 1


async def test_option_flow_subscribe_logs(
    hass: HomeAssistant,
    mock_client: APIClient,
    mock_generic_device_entry: MockGenericDeviceEntryType,
) -> None:
    """Test config flow options with subscribe logs."""
    entry = await mock_generic_device_entry(
        mock_client=mock_client,
    )

    result = await hass.config_entries.options.async_init(entry.entry_id)

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "init"
    assert result["data_schema"]({}) == {
        CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
        CONF_SUBSCRIBE_LOGS: False,
    }

    with patch(
        "homeassistant.components.esphome.async_setup_entry", return_value=True
    ) as mock_reload:
        result = await hass.config_entries.options.async_configure(
            result["flow_id"],
            user_input={CONF_ALLOW_SERVICE_CALLS: False, CONF_SUBSCRIBE_LOGS: True},
        )
        await hass.async_block_till_done()

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_ALLOW_SERVICE_CALLS: False,
        CONF_SUBSCRIBE_LOGS: True,
    }
    assert len(mock_reload.mock_calls) == 1


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_discovers_name_no_dashboard(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test user step can discover the name and the there is not dashboard."""
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(
            uses_password=False,
            name="test",
            mac_address="11:22:33:44:55:AA",
        ),
    ]

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )
    await hass.async_block_till_done()

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "test"}

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: VALID_NOISE_PSK,
        CONF_DEVICE_NAME: "test",
    }
    assert mock_client.noise_psk == VALID_NOISE_PSK


async def mqtt_discovery_test_abort(
    hass: HomeAssistant, payload: str, reason: str
) -> None:
    """Test discovery aborted."""
    service_info = MqttServiceInfo(
        topic="esphome/discover/test",
        payload=payload,
        qos=0,
        retain=False,
        subscribed_topic="esphome/discover/#",
        timestamp=None,
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_MQTT}, data=service_info
    )
    assert flow["type"] is FlowResultType.ABORT
    assert flow["reason"] == reason


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_mqtt_no_mac(hass: HomeAssistant) -> None:
    """Test discovery aborted if mac is missing in MQTT payload."""
    await mqtt_discovery_test_abort(hass, "{}", "mqtt_missing_mac")


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_mqtt_empty_payload(hass: HomeAssistant) -> None:
    """Test discovery aborted if MQTT payload is empty."""
    await mqtt_discovery_test_abort(hass, "", "mqtt_missing_payload")


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_mqtt_no_api(hass: HomeAssistant) -> None:
    """Test discovery aborted if api/port is missing in MQTT payload."""
    await mqtt_discovery_test_abort(hass, '{"mac":"abcdef123456"}', "mqtt_missing_api")


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_mqtt_no_ip(hass: HomeAssistant) -> None:
    """Test discovery aborted if ip is missing in MQTT payload."""
    await mqtt_discovery_test_abort(
        hass, '{"mac":"abcdef123456","port":6053}', "mqtt_missing_ip"
    )


@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
async def test_discovery_mqtt_initiation(hass: HomeAssistant) -> None:
    """Test discovery importing works."""
    service_info = MqttServiceInfo(
        topic="esphome/discover/test",
        payload='{"name":"mock_name","mac":"1122334455aa","port":6053,"ip":"192.168.43.183"}',
        qos=0,
        retain=False,
        subscribed_topic="esphome/discover/#",
        timestamp=None,
    )
    flow = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_MQTT}, data=service_info
    )

    result = await hass.config_entries.flow.async_configure(
        flow["flow_id"], user_input={}
    )

    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test"
    assert result["data"][CONF_HOST] == "192.168.43.183"
    assert result["data"][CONF_PORT] == 6053

    assert result["result"]
    assert result["result"].unique_id == "11:22:33:44:55:aa"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_name_conflict_migrate(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test handle migration on name conflict."""
    existing_entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_DEVICE_NAME: "test"},
        unique_id="11:22:33:44:55:cc",
    )
    existing_entry.add_to_hass(hass)
    device_info = DeviceInfo(
        uses_password=False,
        name="test",
        mac_address="11:22:33:44:55:AA",
    )
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )
    await hass.async_block_till_done()

    assert result["type"] is FlowResultType.MENU
    assert result["step_id"] == "name_conflict"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={"next_step_id": "name_conflict_migrate"}
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "name_conflict_migrated"
    assert result["description_placeholders"] == {
        "existing_mac": "11:22:33:44:55:cc",
        "mac": "11:22:33:44:55:aa",
        "name": "test",
    }
    assert existing_entry.data == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }
    assert existing_entry.unique_id == "11:22:33:44:55:aa"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_name_conflict_overwrite(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test handle overwrite on name conflict."""
    existing_entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_DEVICE_NAME: "test"},
        unique_id="11:22:33:44:55:cc",
    )
    existing_entry.add_to_hass(hass)
    device_info = DeviceInfo(
        uses_password=False,
        name="test",
        mac_address="11:22:33:44:55:AA",
    )
    mock_client.device_info = AsyncMock(return_value=device_info)
    mock_client.list_entities_services = AsyncMock(return_value=([], []))
    mock_client.device_info_and_list_entities = AsyncMock(
        return_value=(device_info, [], [])
    )

    result = await hass.config_entries.flow.async_init(
        DOMAIN,
        context={"source": config_entries.SOURCE_USER},
        data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
    )

    assert result["type"] is FlowResultType.MENU
    assert result["step_id"] == "name_conflict"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={"next_step_id": "name_conflict_overwrite"}
    )
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["data"] == {
        CONF_HOST: "127.0.0.1",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }
    assert result["context"]["unique_id"] == "11:22:33:44:55:aa"


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_success_with_same_ip_new_name(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation with same ip and new name."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_successful"
    assert entry.data[CONF_HOST] == "127.0.0.1"
    assert entry.data[CONF_DEVICE_NAME] == "other"


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_success_with_new_ip_new_name(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation with new ip and new name."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_successful"
    assert entry.data[CONF_HOST] == "127.0.0.2"
    assert entry.data[CONF_DEVICE_NAME] == "other"


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_success_with_new_ip_same_name(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation with new ip and same name."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
            CONF_NOISE_PSK: VALID_NOISE_PSK,
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_successful"
    assert entry.data[CONF_HOST] == "127.0.0.1"
    assert entry.data[CONF_DEVICE_NAME] == "test"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_success_noise_psk_changes(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation with new ip and new noise psk."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
            CONF_NOISE_PSK: VALID_NOISE_PSK,
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)
    mock_client.device_info.side_effect = [
        RequiresEncryptionAPIError,
        InvalidEncryptionKeyAPIError("Wrong key", "test"),
        DeviceInfo(uses_password=False, name="test", mac_address="11:22:33:44:55:aa"),
    ]
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "Mock Title (test)"}
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )

    assert result["type"] is FlowResultType.FORM
    assert result["step_id"] == "encryption_key"
    assert result["description_placeholders"] == {"name": "Mock Title (test)"}
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_successful"
    assert entry.data[CONF_HOST] == "127.0.0.1"
    assert entry.data[CONF_DEVICE_NAME] == "test"
    assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_name_conflict_with_existing_entry(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig with a name conflict with an existing entry."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    entry2 = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.2",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "other",
        },
        unique_id="11:22:33:44:55:bb",
    )
    entry2.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.3", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_name_conflict"
    assert result["description_placeholders"] == {
        "existing_title": "Mock Title",
        "expected_mac": "11:22:33:44:55:aa",
        "host": "127.0.0.3",
        "name": "test",
    }


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_attempt_to_change_mac_aborts(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation with valid PSK attempting to change mac."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="other", mac_address="11:22:33:44:55:bb"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_unique_id_changed"
    assert CONF_NOISE_PSK not in entry.data
    assert result["description_placeholders"] == {
        "expected_mac": "11:22:33:44:55:aa",
        "host": "127.0.0.2",
        "name": "test",
        "unexpected_device_name": "other",
        "unexpected_mac": "11:22:33:44:55:bb",
    }


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_mac_used_by_other_entry(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig when there is another entry for the mac."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)
    entry2 = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.2",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test4",
        },
        unique_id="11:22:33:44:55:bb",
    )
    entry2.add_to_hass(hass)
    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_already_configured"
    assert result["description_placeholders"] == {
        "title": "Mock Title",
        "name": "test4",
        "mac": "11:22:33:44:55:bb",
    }


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_name_conflict_migrate(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation when device has been replaced."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.MENU
    assert result["step_id"] == "name_conflict"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={"next_step_id": "name_conflict_migrate"}
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "name_conflict_migrated"

    assert entry.data == {
        CONF_HOST: "127.0.0.2",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test",
    }
    assert entry.unique_id == "11:22:33:44:55:bb"


@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
async def test_reconfig_name_conflict_overwrite(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test reconfig initiation when device has been replaced."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={
            CONF_HOST: "127.0.0.1",
            CONF_PORT: 6053,
            CONF_PASSWORD: "",
            CONF_DEVICE_NAME: "test",
        },
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    result = await entry.start_reconfigure_flow(hass)

    mock_client.device_info.return_value = DeviceInfo(
        uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
    )
    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
    )

    assert result["type"] is FlowResultType.MENU
    assert result["step_id"] == "name_conflict"

    result = await hass.config_entries.flow.async_configure(
        result["flow_id"], user_input={"next_step_id": "name_conflict_overwrite"}
    )
    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "reconfigure_successful"

    assert (
        hass.config_entries.async_entry_for_domain_unique_id(
            DOMAIN, "11:22:33:44:55:bb"
        )
        is not None
    )
    assert (
        hass.config_entries.async_entry_for_domain_unique_id(
            DOMAIN, "11:22:33:44:55:aa"
        )
        is None
    )


@pytest.mark.usefixtures("mock_setup_entry")
async def test_discovery_dhcp_no_probe_same_host_port_none(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test dhcp discovery does not probe when host matches and port is None."""
    entry = MockConfigEntry(
        domain=DOMAIN,
        data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
        unique_id="11:22:33:44:55:aa",
    )
    entry.add_to_hass(hass)

    # DHCP discovery with same MAC and host (WiFi device)
    service_info = DhcpServiceInfo(
        ip="192.168.43.183",
        hostname="test8266",
        macaddress="1122334455aa",  # Same MAC as configured
    )

    result = await hass.config_entries.flow.async_init(
        DOMAIN, context={"source": config_entries.SOURCE_DHCP}, data=service_info
    )

    assert result["type"] is FlowResultType.ABORT
    assert result["reason"] == "already_configured"

    # Verify device_info was NOT called (no probing)
    mock_client.device_info.assert_not_called()

    # Host should remain unchanged
    assert entry.data[CONF_HOST] == "192.168.43.183"


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_starts_zwave_discovery(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test that the user flow starts Z-Wave JS discovery when device has Z-Wave capabilities."""
    # Mock device with Z-Wave capabilities
    mock_client.device_info = AsyncMock(
        return_value=DeviceInfo(
            uses_password=False,
            name="test-zwave-device",
            mac_address="11:22:33:44:55:BB",
            zwave_proxy_feature_flags=1,
            zwave_home_id=1234567890,
        )
    )
    mock_client.connected_address = "mock-connected-address"

    # Track flow.async_init calls and async_get calls
    original_async_init = hass.config_entries.flow.async_init
    original_async_get = hass.config_entries.flow.async_get
    flow_init_calls = []
    zwave_flow_id = "mock-zwave-flow-id"

    async def track_async_init(*args, **kwargs):
        flow_init_calls.append((args, kwargs))
        # For the Z-Wave flow, return a mock result with the flow_id
        if args and args[0] == "zwave_js":
            return {"flow_id": zwave_flow_id, "type": FlowResultType.FORM}
        # Otherwise call the original
        return await original_async_init(*args, **kwargs)

    def mock_async_get(flow_id: str):
        # Return a mock flow for the Z-Wave flow_id
        if flow_id == zwave_flow_id:
            return MagicMock()
        return original_async_get(flow_id)

    with (
        patch.object(
            hass.config_entries.flow, "async_init", side_effect=track_async_init
        ),
        patch.object(hass.config_entries.flow, "async_get", side_effect=mock_async_get),
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "192.168.1.100", CONF_PORT: 6053},
        )

    # Verify the entry was created
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test-zwave-device"
    assert result["data"] == {
        CONF_HOST: "192.168.1.100",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test-zwave-device",
    }

    # First call is ESPHome flow, second should be Z-Wave flow
    assert len(flow_init_calls) == 2
    zwave_call_args, zwave_call_kwargs = flow_init_calls[1]
    assert zwave_call_args[0] == "zwave_js"
    assert zwave_call_kwargs["context"] == {
        "source": config_entries.SOURCE_ESPHOME,
        "discovery_key": discovery_flow.DiscoveryKey(
            domain="esphome", key="11:22:33:44:55:BB", version=1
        ),
    }
    assert zwave_call_kwargs["data"] == ESPHomeServiceInfo(
        name="test-zwave-device",
        zwave_home_id=1234567890,
        ip_address="mock-connected-address",
        port=6053,
        noise_psk=None,
    )

    # Verify next_flow was set
    assert result["next_flow"] == (config_entries.FlowType.CONFIG_FLOW, zwave_flow_id)


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_no_zwave_discovery_without_home_id(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test that the user flow does not start Z-Wave JS discovery when zwave_home_id is not set."""
    # Mock device with Z-Wave capabilities but no home ID
    mock_client.device_info = AsyncMock(
        return_value=DeviceInfo(
            uses_password=False,
            name="test-zwave-device-no-id",
            mac_address="11:22:33:44:55:CC",
            zwave_proxy_feature_flags=1,
            zwave_home_id=0,  # No home ID set (fresh adapter or unplugged)
        )
    )
    mock_client.connected_address = "192.168.1.103"

    # Track flow.async_init calls
    original_async_init = hass.config_entries.flow.async_init
    flow_init_calls = []

    async def track_async_init(*args, **kwargs):
        flow_init_calls.append((args, kwargs))
        return await original_async_init(*args, **kwargs)

    with patch.object(
        hass.config_entries.flow, "async_init", side_effect=track_async_init
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "192.168.1.103", CONF_PORT: 6053},
        )

    # Verify the ESPHome entry was created
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test-zwave-device-no-id"
    assert result["data"] == {
        CONF_HOST: "192.168.1.103",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test-zwave-device-no-id",
    }

    # Verify only ESPHome flow was initiated, no Z-Wave flow
    assert len(flow_init_calls) == 1
    assert flow_init_calls[0][0][0] == DOMAIN

    # Verify next_flow was NOT set
    assert "next_flow" not in result


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_no_zwave_discovery_without_capabilities(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test that the user flow does not start Z-Wave JS discovery when device has no Z-Wave capabilities."""
    # Mock device without Z-Wave capabilities
    mock_client.device_info = AsyncMock(
        return_value=DeviceInfo(
            uses_password=False,
            name="test-regular-device",
            mac_address="11:22:33:44:55:CC",
        )
    )

    # Track flow.async_init calls
    original_async_init = hass.config_entries.flow.async_init
    flow_init_calls = []

    async def track_async_init(*args, **kwargs):
        flow_init_calls.append((args, kwargs))
        return await original_async_init(*args, **kwargs)

    with patch.object(
        hass.config_entries.flow, "async_init", side_effect=track_async_init
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "192.168.1.101", CONF_PORT: 6053},
        )

    # Verify the entry was created
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test-regular-device"

    # Verify Z-Wave discovery flow was NOT started (only ESPHome flow)
    assert len(flow_init_calls) == 1

    # Verify next_flow was not set
    assert "next_flow" not in result


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_user_flow_zwave_discovery_aborts(
    hass: HomeAssistant, mock_client: APIClient
) -> None:
    """Test that the user flow handles Z-Wave discovery abort gracefully."""
    # Mock device with Z-Wave capabilities
    mock_client.device_info = AsyncMock(
        return_value=DeviceInfo(
            uses_password=False,
            name="test-zwave-device",
            mac_address="11:22:33:44:55:DD",
            zwave_proxy_feature_flags=1,
            zwave_home_id=9876543210,
        )
    )
    mock_client.connected_address = "192.168.1.102"

    # Track flow.async_init calls
    original_async_init = hass.config_entries.flow.async_init
    flow_init_calls = []

    async def track_async_init(*args, **kwargs):
        flow_init_calls.append((args, kwargs))
        # For the Z-Wave flow, return an ABORT result
        if args and args[0] == "zwave_js":
            return {
                "type": FlowResultType.ABORT,
                "reason": "already_configured",
            }
        # Otherwise call the original
        return await original_async_init(*args, **kwargs)

    with patch.object(
        hass.config_entries.flow, "async_init", side_effect=track_async_init
    ):
        result = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_USER},
            data={CONF_HOST: "192.168.1.102", CONF_PORT: 6053},
        )

    # Verify the ESPHome entry was still created despite Z-Wave flow aborting
    assert result["type"] is FlowResultType.CREATE_ENTRY
    assert result["title"] == "test-zwave-device"
    assert result["data"] == {
        CONF_HOST: "192.168.1.102",
        CONF_PORT: 6053,
        CONF_PASSWORD: "",
        CONF_NOISE_PSK: "",
        CONF_DEVICE_NAME: "test-zwave-device",
    }

    # Verify Z-Wave discovery flow was attempted
    assert len(flow_init_calls) == 2
    zwave_call_args, zwave_call_kwargs = flow_init_calls[1]
    assert zwave_call_args[0] == "zwave_js"
    assert zwave_call_kwargs["context"]["source"] == config_entries.SOURCE_ESPHOME
    assert zwave_call_kwargs["context"]["discovery_key"] == discovery_flow.DiscoveryKey(
        domain=DOMAIN,
        key="11:22:33:44:55:DD",
        version=1,
    )
    assert zwave_call_kwargs["data"] == ESPHomeServiceInfo(
        name="test-zwave-device",
        zwave_home_id=9876543210,
        ip_address="192.168.1.102",
        port=6053,
        noise_psk=None,
    )

    # Verify next_flow was NOT set since Z-Wave flow aborted
    assert "next_flow" not in result


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_zeroconf_notifies_improv_ble(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test that zeroconf discovery notifies improv_ble integration."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "aabbccddeeff",
        },
        type="mock_type",
    )

    # Patch improv_ble to ensure it's available and track calls
    with patch(
        "homeassistant.components.improv_ble.async_register_next_flow"
    ) as mock_register:
        flow = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_ZEROCONF},
            data=service_info,
        )

    assert flow["type"] is FlowResultType.FORM
    assert flow["step_id"] == "discovery_confirm"

    # Verify improv_ble.async_register_next_flow was called with correct parameters
    assert len(mock_register.mock_calls) == 1
    call_args = mock_register.mock_calls[0].args
    assert call_args[0] is hass  # HomeAssistant instance
    # WiFi MAC aabbccddeeff + 1 = Bluetooth MAC aabbccddee00
    # (wifi_mac_to_bluetooth_mac from aioesphomeapi)
    expected_ble_mac = wifi_mac_to_bluetooth_mac("aa:bb:cc:dd:ee:ff")
    assert call_args[1] == expected_ble_mac  # BLE MAC address
    assert call_args[2] == flow["flow_id"]  # Flow ID


@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
async def test_zeroconf_when_improv_ble_not_available(
    hass: HomeAssistant,
    mock_client: APIClient,
) -> None:
    """Test that zeroconf discovery works when improv_ble is not available."""
    service_info = ZeroconfServiceInfo(
        ip_address=ip_address("192.168.43.183"),
        ip_addresses=[ip_address("192.168.43.183")],
        hostname="test8266.local.",
        name="mock_name",
        port=6053,
        properties={
            "mac": "aabbccddeeff",
        },
        type="mock_type",
    )

    # Mock async_import_module to return None (simulating improv_ble not available)
    with patch(
        "homeassistant.components.esphome.config_flow.async_import_module",
        return_value=None,
    ):
        flow = await hass.config_entries.flow.async_init(
            DOMAIN,
            context={"source": config_entries.SOURCE_ZEROCONF},
            data=service_info,
        )

    # Flow should still work even without improv_ble
    assert flow["type"] is FlowResultType.FORM
    assert flow["step_id"] == "discovery_confirm"
